"""
Copyright (c) 2022 Huawei Technologies Co.,Ltd.

openGauss is licensed under Mulan PSL v2.
You can use this software according to the terms and conditions of the Mulan PSL v2.
You may obtain a copy of Mulan PSL v2 at:

          http://license.coscl.org.cn/MulanPSL2

THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND,
EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT,
MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE.
See the Mulan PSL v2 for more details.
"""
"""
Case Type   : 故障&可靠性测试
Case Name   : 备机表空间目录权限不同，表空间进行建表
Create At   : 2022/05/18
Owner       : @peilinqian
Description :
    1、创建表空间;
    2、修改表空间目录权限为000，进行表创建;
    3、恢复集群，删除表;
    4、修改表空间目录权限为100，进行表创建;
    5、恢复集群，删除表;
    6、修改表空间目录权限为200，进行表创建;
    7、恢复集群，删除表;
    8、修改表空间目录权限为300，进行表创建;
    9、恢复集群，删除表;
    10、修改表空间目录权限为400，进行表创建;
    11、恢复集群，删除表;
    12、修改表空间目录权限为500，进行表创建;
    13、恢复集群，删除表;
    14、修改表空间目录权限为600，进行表创建;
    15、恢复集群，删除表;
Expect      :
    1、创建表空间;创建成功
    2、修改表空间目录权限为000，进行表创建;创建成功，备机异常
    3、恢复集群，删除表; 成功
    4、修改表空间目录权限为100，进行表创建;创建成功
    5、恢复集群，删除表; 成功
    6、修改表空间目录权限为200，进行表创建;创建成功，备机异常
    7、恢复集群，删除表; 成功
    8、修改表空间目录权限为300，进行表创建;创建成功
    9、恢复集群，删除表; 成功
    10、修改表空间目录权限为400，进行表创建;创建成功，备机异常
    11、恢复集群，删除表; 成功
    12、修改表空间目录权限为500，进行表创建;创建成功
    13、恢复集群，删除表; 成功
    14、修改表空间目录权限为600，进行表创建;创建成功，备机异常
    15、恢复集群，删除表; 成功
History     :
"""

import os
import re
import unittest

from testcase.utils.Common import Common
from testcase.utils.CommonSH import CommonSH
from testcase.utils.CommonSH import RestartDbCluster
from testcase.utils.Constant import Constant
from testcase.utils.Logger import Logger
from yat.test import Node
from yat.test import macro


class DbSysCase274(unittest.TestCase):
    nodes_tuple = ('PrimaryDbUser', 'Standby1DbUser', 'Standby2DbUser')

    @RestartDbCluster(*nodes_tuple)
    def setUp(self):
        self.log = Logger()
        self.log.info(f'-----{os.path.basename(__file__)} start-----')
        self.pri_dbuser = Node(node='PrimaryDbUser')
        self.sta1_root = Node(node='Standby1Root')
        self.sta1_dbuser = Node(node='Standby1DbUser')
        self.sta2_dbuser = Node(node='Standby2DbUser')
        self.pri_sh = CommonSH('PrimaryDbUser')
        self.constant = Constant()
        self.com = Common()
        self.tbspc_name = 'tbspc_dbsys274'
        self.tbspc_path = os.path.join(os.path.dirname(
            macro.DB_INSTANCE_PATH), 'tbspc_dbsys274')
        self.create_ts_sql = f"create tablespace {self.tbspc_name} " \
            f"location '{self.tbspc_path}';"
        self.drop_ts_sql = f"drop tablespace if exists {self.tbspc_name};"
        self.table_name = 't_dbsys274'
        self.create_tb_sql = f"create table {self.table_name} " \
            f"(id int,name varchar(100)) tablespace {self.tbspc_name};"
        self.drop_tb_sql = f"drop table if exists {self.table_name};" \
            f"checkpoint"
        self.err_msg = 'ERROR:  could not stat directory .*: ' \
                       'Permission denied'
        self.chmod_cmd = f'chmod authority {self.tbspc_path} &&' \
            f'ls -al {os.path.dirname(self.tbspc_path)} ' \
            f'|grep {self.tbspc_name}'

    def test_main(self):
        step_txt = '----step1:创建表空间;expect:创建成功----'
        self.log.info(step_txt)
        create_result = self.pri_sh.executDbSql(self.create_ts_sql)
        self.log.info(create_result)
        self.assertIn(self.constant.CREATE_TBLSPACE_SUCCESS_MSG,
                      create_result, "执行失败" + step_txt)

        step_txt = '----step2:修改表空间目录权限为000，进行表创建;expect:创建成功，备机异常----'
        self.log.info(step_txt)
        chmod_cmd = self.chmod_cmd.replace('authority', '000')
        chmod_result = self.com.get_sh_result(self.sta1_root, chmod_cmd)
        self.assertIn('d---------', chmod_result, "执行失败" + step_txt)
        create_result = self.pri_sh.executDbSql(self.create_tb_sql)
        self.log.info(create_result)
        self.assertIn(self.constant.TABLE_CREATE_SUCCESS, create_result,
                      "执行失败" + step_txt)
        status = self.pri_sh.getDbClusterStatus('status')
        self.assertFalse(status, "执行失败" + step_txt)

        step_txt = '----step3:恢复集群，删除表; expect:成功----'
        self.log.info(step_txt)
        recovery_msg = self.recovery()
        self.assertTrue(recovery_msg, "执行失败" + step_txt)

        step_txt = '----step4:修改表空间目录权限为100，进行表创建;expect:创建成功----'
        self.log.info(step_txt)
        chmod_cmd = self.chmod_cmd.replace('authority', '100')
        chmod_result = self.com.get_sh_result(self.sta1_root, chmod_cmd)
        self.assertIn('d--x------', chmod_result, "执行失败" + step_txt)
        create_result = self.pri_sh.executDbSql(self.create_tb_sql)
        self.log.info(create_result)
        self.assertIn(self.constant.TABLE_CREATE_SUCCESS, create_result,
                      "执行失败" + step_txt)
        status = self.pri_sh.getDbClusterStatus('status')
        self.assertTrue(status, "执行失败" + step_txt)

        step_txt = '----step5:恢复集群，删除表; expect:成功----'
        self.log.info(step_txt)
        recovery_msg = self.recovery()
        self.assertTrue(recovery_msg, "执行失败" + step_txt)

        step_txt = '----step6:修改表空间目录权限为200，进行表创建;expect:创建成功，备机异常----'
        self.log.info(step_txt)
        chmod_cmd = self.chmod_cmd.replace('authority', '200')
        chmod_result = self.com.get_sh_result(self.sta1_root, chmod_cmd)
        self.assertIn('d-w-------', chmod_result, "执行失败" + step_txt)
        create_result = self.pri_sh.executDbSql(self.create_tb_sql)
        self.log.info(create_result)
        self.assertIsNotNone(re.findall(self.err_msg, create_result),
                             "执行失败" + step_txt)
        status = self.pri_sh.getDbClusterStatus('status')
        self.assertFalse(status, "执行失败" + step_txt)

        step_txt = '----step7:恢复集群，删除表; expect:成功----'
        self.log.info(step_txt)
        recovery_msg = self.recovery()
        self.assertTrue(recovery_msg, "执行失败" + step_txt)

        step_txt = '----step8:修改表空间目录权限为300，进行表创建;expect:创建成功----'
        self.log.info(step_txt)
        chmod_cmd = self.chmod_cmd.replace('authority', '300')
        chmod_result = self.com.get_sh_result(self.sta1_root, chmod_cmd)
        self.assertIn('d-wx------', chmod_result, "执行失败" + step_txt)
        create_result = self.pri_sh.executDbSql(self.create_tb_sql)
        self.log.info(create_result)
        self.assertIn(self.constant.TABLE_CREATE_SUCCESS, create_result,
                      "执行失败" + step_txt)

        step_txt = '----step9:恢复集群，删除表; expect:成功----'
        self.log.info(step_txt)
        recovery_msg = self.recovery()
        self.assertTrue(recovery_msg, "执行失败" + step_txt)

        step_txt = '----step10:修改表空间目录权限为400，进行表创建;expect:创建成功，备机异常----'
        self.log.info(step_txt)
        chmod_cmd = self.chmod_cmd.replace('authority', '400')
        chmod_result = self.com.get_sh_result(self.sta1_root, chmod_cmd)
        self.assertIn('dr--------', chmod_result, "执行失败" + step_txt)
        create_result = self.pri_sh.executDbSql(self.create_tb_sql)
        self.log.info(create_result)
        self.assertIsNotNone(re.findall(self.err_msg, create_result),
                             "执行失败" + step_txt)
        status = self.pri_sh.getDbClusterStatus('status')
        self.assertFalse(status, "执行失败" + step_txt)

        step_txt = '----step11:恢复集群，删除表; expect:成功----'
        self.log.info(step_txt)
        recovery_msg = self.recovery()
        self.assertTrue(recovery_msg, "执行失败" + step_txt)

        step_txt = '----step12:修改表空间目录权限为500，进行表创建;expect:创建成功----'
        self.log.info(step_txt)
        chmod_cmd = self.chmod_cmd.replace('authority', '500')
        chmod_result = self.com.get_sh_result(self.sta1_root, chmod_cmd)
        self.assertIn('dr-x------', chmod_result, "执行失败" + step_txt)
        create_result = self.pri_sh.executDbSql(self.create_tb_sql)
        self.log.info(create_result)
        self.assertIn(self.constant.TABLE_CREATE_SUCCESS, create_result,
                      "执行失败" + step_txt)

        step_txt = '----step13:恢复集群，删除表; expect:成功----'
        self.log.info(step_txt)
        recovery_msg = self.recovery()
        self.assertTrue(recovery_msg, "执行失败" + step_txt)

        step_txt = '----step14:修改表空间目录权限为600，进行表创建;expect:创建成功，备机异常----'
        self.log.info(step_txt)
        chmod_cmd = self.chmod_cmd.replace('authority', '600')
        chmod_result = self.com.get_sh_result(self.sta1_root, chmod_cmd)
        self.assertIn('drw-------', chmod_result, "执行失败" + step_txt)
        create_result = self.pri_sh.executDbSql(self.create_tb_sql)
        self.log.info(create_result)
        self.assertIsNotNone(re.findall(self.err_msg, create_result),
                             "执行失败" + step_txt)
        status = self.pri_sh.getDbClusterStatus('status')
        self.assertFalse(status, "执行失败" + step_txt)

        step_txt = '----step15:恢复集群，删除表; expect:成功----'
        self.log.info(step_txt)
        recovery_msg = self.recovery()
        self.assertTrue(recovery_msg, "执行失败" + step_txt)

    def recovery(self):
        self.log.info("--恢复集群--")
        chmod_cmd = self.chmod_cmd.replace('authority', '700')
        self.com.get_sh_result(self.sta1_root, chmod_cmd)
        start_result = self.pri_sh.startDbCluster()
        self.log.info("--删除表--")
        create_msg = self.pri_sh.executDbSql(self.drop_tb_sql)
        self.log.info(create_msg)
        if start_result and self.constant.TABLE_DROP_SUCCESS in create_msg:
            return True
        else:
            return False

    def tearDown(self):
        self.log.info("----this is tearDown----")
        step1_txt = '----清理表空间; expect:成功----'
        self.log.info(step1_txt)
        chmod_cmd = self.chmod_cmd.replace('authority', '700')
        chmod_result = self.com.get_sh_result(self.sta1_root, chmod_cmd)
        drop_tb_result = self.pri_sh.executDbSql(self.drop_tb_sql)
        self.log.info(drop_tb_result)
        drop_ts_result = self.pri_sh.executDbSql(self.drop_ts_sql)
        self.log.info(drop_ts_result)
        step2_txt = '----删除表空间路径; expect:成功----'
        self.log.info(step2_txt)
        del_cmd = f"rm -rf {self.tbspc_path}; " \
            f"if [ -d {self.tbspc_path} ]; then echo 'exists'; " \
            f"else echo 'not exists'; fi;"
        self.log.info(del_cmd)
        del_result1 = self.pri_dbuser.sh(del_cmd).result()
        del_result2 = self.sta1_dbuser.sh(del_cmd).result()
        del_result3 = self.sta2_dbuser.sh(del_cmd).result()
        self.log.info(del_result1)
        self.log.info(del_result2)
        self.log.info(del_result3)

        self.log.info(f'-----{os.path.basename(__file__)} end-----')
        self.assertIn('drwx------', chmod_result, "执行失败" + step1_txt)
        self.assertIn(self.constant.TABLE_DROP_SUCCESS, drop_tb_result,
                      "执行失败" + step1_txt)
        self.assertIn(self.constant.drop_tablespace_success_msg,
                      drop_ts_result, "执行失败" + step1_txt)
        self.assertEqual('not exists', del_result1, "执行失败" + step2_txt)
        self.assertEqual('not exists', del_result2, "执行失败" + step2_txt)
        self.assertEqual('not exists', del_result3, "执行失败" + step2_txt)
